## Modelado y Síntesis de Sistemas Electrónico Digitales

# Apartado 4. Implementación del diseño completo



## Realizada por:

Guillermo Alba Sánchez Adrián García-Vera de Lope

## Índice

### Apartado 4. Implementación del diseño completo

| Índice                                        | 1  |
|-----------------------------------------------|----|
| Introducción                                  | 2  |
| Modelar la entidad top_system                 | 3  |
| top_system                                    | 3  |
| 1. epp_controller                             | 5  |
| 2. decoder_epp                                | 5  |
| 3. codec_controller                           | 5  |
| Banco de pruebas de la entidad top_system     | 6  |
| top_system_tb                                 | 6  |
| 1. epp_device_ts                              | 6  |
| 2. LM4550                                     | 8  |
| Simulación funcional de la entidad top_system | 12 |
| Simulación temporal de la entidad top_system  | 14 |
| Salida MatLab                                 | 15 |
| Informe sobre los recursos utilizados         | 15 |

#### Introducción

Este apartado es la parte final de la práctica libre, por lo que el objetivo principal es conseguir la unión de todos los componentes y su implementación final en la placa Atlys de Digilent a través del CODEC de audio LM4550.

En la figura 1 podemos apreciar que la entidad *top\_system* se compone de las entidades *codec\_controller*, *decoder\_epp*, y *epp\_controller*, todas ellas modeladas en los apartados anteriores de la práctica libre.



Figura 1. top\_system

Para conseguir el completo funcionamiento del sistema se debe poder generar una señal de audio con el codec LM4550. Esta señal tendrá unos valores de frecuencia y amplitud que se transmiten a la FPGA de la placa Atlys a través del puerto USB del ordenador.

Como vemos en la figura 2, las datos enviados por el PC son convertidos por el driver CY7C68013A, que está en la placa Atlys y convierte los datos recibidos a un formato parallelo (EPP- Enhanced Parallel Port), enviandolos a la FPGA.



Figura 2. Conexión del hardware a utilizar

#### Modelar la entidad top\_system

Antes de hablar sobre las interconexiones de los componentes principales del sistema, haremos una introducción de aquellos componentes que se nos han dado y que usamos por debajo de estos principales:

#### top system

El modelado de este sistema ha sido muy sencillo ya que sólo quedaba unir los distintos componentes, para ello tenemos que conectar las entradas del *top\_system* al *epp\_controller* (ASTRB, DSTRB, DATA, PWRITE, PWAIT), las salidas al codec\_controller (RESET, SYNC, SDATA\_OUT) además de las señales de sincronización (RELOJ y RST).

Visto esto tan solo nos faltaría "interconectar" los componentes (señales de las líneas 24-33), para ello usaremos señales auxiliares que nos permiten conectar los componentes que las tienen como salida y en los que la tienen como entrada, parece difícil pero dejamos muestras del código utilizado para que se vea con mayor claridad en las Figuras 3 y 4.

```
entity top_system is
 port (
   RELOJ
             : in std logic;
   RST
             : in std_logic;
   ASTRB
             : in std_logic;
             : in std_logic;
   DSTRB
             : in std_logic_vector(7 downto 0);
   DATA
   PWRITE
             : in std_logic;
   PWAIT
             : out std_logic;
   BIT_CLK
             : in std_logic;
   RESET
             : out std_logic;
   SYNC
             : out std_logic;
   SDATA_OUT : out std_logic);
end;
architecture rtl of top_system is
 signal DIR_AUX
                      : std_logic_vector(7 downto 0);
                      : std_logic_vector(7 downto 0);
 signal DATO_AUX
  signal DATOS_VLD_AUX : std_logic;
  signal RESTART_AUX
                          : std_logic;
 signal VOL_CODE_AUX
                          : std_logic_vector(4 downto 0);
  signal FREC_CODE_AUX
                          : std_logic_vector(7 downto 0);
  signal CHANNEL_AUX
                         : std_logic_vector(1 downto 0);
```

Figura 3. Señales del top\_system

```
begin
  epp : entity work.epp_controller
    port map (
      CLK
                ⇒ RELOJ,
      RST
                ⇒ RST,
      ASTRB
                ⇒ ASTRB,
      DSTRB
                ⇒ DSTRB,
      DATA
                => DATA,
      PWRITE
                ⇒ PWRITE,
      PWAIT
                => PWAIT,
      DIR
                ⇒ DIR_AUX,
      DATO
                ⇒ DATO_AUX,
      DATOS_VLD => DATOS_VLD_AUX
    );
  decoder : entity work.decoder_epp
    port map (
      CLK
                ⇒ RELOJ,
      RST
                => RST,
      DIR
                ⇒ DIR_AUX,
      DATO
                ⇒ DATO_AUX,
      DATOS_VLD => DATOS_VLD_AUX,
      RESTART
                ⇒ RESTART_AUX,
      VOL_CODE -> VOL_CODE_AUX,
      FREC_CODE => FREC_CODE_AUX,
      CHANNEL
               -> CHANNEL_AUX
    );
  codec : entity work.codec_controller
    port map (
      RST
                => RST,
      BIT_CLK
                ⇒ BIT_CLK,
      CLK
                ⇒ RELOJ,
      VOL_CODE => VOL_CODE_AUX,
      RESTART
                ⇒ RESTART_AUX,
                ⇒ CHANNEL_AUX,
      CHANNEL
                ⇒ RESET,
      RESET
                => SYNC,
      SYNC
      FREC_CODE => FREC_CODE_AUX,
      SDATA_OUT => SDATA_OUT
    );
end rtl;
```

Figura 4. Conexión de los componentes con las señales

#### 1. epp controller

Componente que modelamos en el Apartado 1, este es un controlador de puerto paralelo que recibe datos a través del cable usb que conectamos a nuestro ordenador y mediante el software Atlantis podremos interactuar con la placa.

Lo principal de este componente es que nos proporciona las señales **dir** y **dato**, que usaremos posteriormente en el componente *decoder\_epp*, para más detalles sobre su funcionamiento podríamos usar la memoria de dicho apartado.

#### 2. decoder epp

Componente que modelamos en el Apartado 3, su función consiste en decodificar las señales dir y dato que se nos proporcionan a través de la entidad *epp\_controller* y proporcionar a la entidad *codec\_controller* las señales necesarias (**restart**, **vol\_code**, **frec\_code** y **channel**). Para recordar mejor su funcionamiento adjuntamos la Tabla 1, en la que podemos observar el comportamiento del sistema según los valores de **dir** y **dato**, además debemos recordar que cuando estos cambiaban, se mandaba también la señal **datos\_vld** = '1', lo cual nos asegura la integridad de los datos enviados.

| Dirección | Dato                | Función                                   |
|-----------|---------------------|-------------------------------------------|
| 11x       | 11x                 | Inicialización del códec LM4550 (RESTART) |
| F0x       | 00x-FFx             | Frecuencias de la nota                    |
| B0x       | 00x-1Fx             | Volumen de la señal de audio              |
| CAx       | DDx                 | Tono por el canal derecho                 |
| CAx       | 11x                 | Tono por el canal izquierdo               |
| CAx       | 22x                 | Tono por ambos canales                    |
| CAx       | Resto combinaciones | Tono por ningún canal                     |

Tabla 1. Direcciones y datos a utilizar por la entidad decoder epp

#### 3. codec controller

Componente que modelamos en el Apartado 2, su función es comunicarse con el códec LM4550, básicamente podríamos afirmar que es una interfaz por encima de este, toma los datos que se pasan por el *decoder\_epp* (**restart**, **vol\_code**, **frec\_code** y **channel**) para producir una salida de audio acorde a esas señales.

Merece la pena recordar también que este componente hace uso de una memoria con 4096 posiciones (señal sinusoidal) a la que accedemos según la señal **frec\_code**.

#### Banco de pruebas de la entidad top\_system

#### top system tb

El funcionamiento es sencillo, se instancian los componentes que veremos a continuación. Se generan las señales de **rst** y **reloj** y se escribe la salida en "*vout.dat*" (que veremos posteriormente, con mayor detalle) como mostramos en la Figura 5.

Figura 5. Escritura de salida de los canales en el fichero vout.dat

#### 1. epp\_device\_ts

Este componente se nos proporciona, pero merece la pena analizar su funcionamiento para poder entender el conjunto del sistema.

¡Atención! Las siguientes capturas están desordenadas como se puede observar en las líneas, pero se ponen en este orden para facilitar la comprensión del código.

El código es bastante sencillo, lo principal es el archivo de datos que leemos ("datos2.dat") como se puede apreciar en la Figura 6 y la llamada recursiva al procedimiento epp\_cycle, que veremos a continuación, el resto son inicializaciones de las señales, algo propio de los bancos de pruebas.

```
file arch_in : text open read_mode is "../SOURCES/datos2.dat";
   variable bf
                 : line:
   variable dato : std_logic_vector(7 downto 0);
   variable dir : std_logic_vector(7 downto 0);
   variable tiempo : time;
begin
  data <= (others => '0');
  PWRITE <= '1';
 DSTRB <= '1';
 ASTRB <= '1';
 dir := (others => '0');
 wait for 330 ns;
 while not endfile(arch_in) loop
    readline (arch_in, bf);
   hread (bf, dir);
    hread (bf, dato);
   read (bf, tiempo);
   epp_cycle ( address => dir,
                data_out => dato);
    wait for tiempo;
  end loop;
  report "FIN CONTROLADO DE LA SIMULACION" severity failure;
 end process;
```

Figura 6. Main del TB del epp device ts

Al no disponer de una placa para trabajar en casa, con este banco de pruebas podremos hacer las simulaciones pertinentes y verificar que nuestro sistema funciona perfectamente antes de realizar la descarga en placa, en él se define un procedimiento que lee del fichero "datos2.dat" hasta que llega al final de este.

Vemos en la Figura 7 que desde las líneas 27 a la 34 tenemos escrituras en la dirección poniendo las señales **PWRITE** y **ASTRB** a '0' y transmitiendo el valor de la señal **data** a la dirección. Desde las líneas 37 a la 44 tenemos un proceso de escritura del dato poniendo las señales **PWRITE** y **DSTRB** a '0' y transmitiendo el valor de la señal **data** al dato.

```
procedure epp_cycle ( address : in std_logic_vector(7 downto 0);
                      data_out : in std_logic_vector(7 downto 0)) is
begin
    PWRITE <= '0';
    wait for 33 ns;
    ASTRB <= '0';
          address;
    wait for 133 ns;
    ASTRB <= '1';
    wait for 33 ns;
          <= (others => 'Z');
    data
    PWRITE <= '1';
    wait for 133 ns;
    PWRITE <= '0';
           data_out;
    data
    wait for 33 ns;
    DSTRB <= '0';
    wait for 133 ns;
    DSTRB <= '1';
    wait for 33 ns;
    data
          <= (others => 'Z');
    PWRITE <= '1';
end procedure;
```

Figura 7. Procedimiento epp\_cycle dentro del epp\_device\_ts

#### 2. LM4550

Quizá deberíamos haber hablado de primeras sobre el códec para poder entender con mayor claridad por qué hemos realizado así el sistema, este archivo se nos proporciona y es sin duda el que más complejidad tiene ya que es el core de nuestro sistema.

Como comentamos anteriormente la interfaz que controla este códec es el *codec\_controller*, iremos descendiendo en el diseño para entender el funcionamiento, para ello podemos ver que al codec\_controller se le conectan las señales **RESTART**, **VOL\_CODE**, **FREC\_CODE** y **CHANNEL**. La señales **RESTART** y **FREC\_CODE** las usaremos directamente en el *codec\_controller* y "no afectan", por tanto nos quedamos con las otros dos.

**VOL\_CODE** es el encargado de modificar el volumen de la salida, esto se realiza mediante el registro Master Volume (0x02) y el PCM Out Volumen (0x18), podemos ver cómo funciona esta parte con el siguiente código (Figura 8):

```
master_vol_r <= REG_02(4 downto 0);
master_vol_l <= REG_02(12 downto 8);
mute_master <= REG_02(15);

process (master_vol_r, master_vol_l, mute_master) is
begin
   if mute_master = '1' then
        master_gn_r <= 0.0;
        master_gn_l <= 0.0;
   else
        master_gn_r <= 10.0**((0.0-real(to_integer(unsigned(master_vol_r)))*1.5)/20.0);
        master_gn_l <= 10.0**((0.0-real(to_integer(unsigned(master_vol_l)))*1.5)/20.0);
   end if;
end process;</pre>
```

Figura 8. Registro 0x20 Master Volume

En la figura 10, lo primero que hacemos es separar los bits del registro PCM Out Volume según la figura 9. Así que, usamos los bits GR[4:0] para controlar la ganancia o atenuación del canal derecho (**out\_vol\_r**), y los bits GL[4:0] controlando el canal izquierdo (**out\_vol\_l**). El último bit controla el valor mute, silenciando las salidas de los dos DACs.

El proceso comprueba si los canales están silenciados (valor de **mute\_out** a '1'), y en caso contrario se ajusta la ganancia o atenuación de los canales izquierdo y derecho en el rango de 12 dB y en pasos de 1.5 dB

| D15  | D14 | D13 | D12 | D11 | D10 | D9  | D8  | D7 | D6 | D5 | D4  | D3  | D2  | D1  | D0  |
|------|-----|-----|-----|-----|-----|-----|-----|----|----|----|-----|-----|-----|-----|-----|
| Mute | 0   | 0   | GL4 | GL3 | GL2 | GL1 | GL0 | 0  | 0  | 0  | GR4 | GR3 | GR2 | GR1 | GR0 |

Figura 9. Funcionalidad de los bits del registro PCM Out Volume

Comentar que las últimas líneas, de pop en adelante son "innecesarias" en el sentido de que pop por nuestra configuración no cambia como comentaremos posteriormente, por tanto los valores de **output\_r** y **stereo\_mix\_r** serán los mismos, lo mismo sucede con el canal izquierdo.

```
dac_out_r <= real(to_integer(signed(dac_in_r(19 downto 2))))*2.0/262144.0;</pre>
dac_out_l <= real(to_integer(signed(dac_in_l(19 downto 2))))*2.0/262144.0;</pre>
out_vol_r <= REG_18(4 downto 0);</pre>
out_vol_l <= REG_18(12 downto 8);</pre>
mute_out <= REG_18(15);
process (out_vol_r, out_vol_l, mute_out) is
begin
  if mute_out = '1' then
    out_gn_r <= 0.0;
    out_gn_l <= 0.0;
  else
    out_gn_r \leftarrow 10.0**((12.0-real(to_integer(unsigned(out_vol_r)))*1.5)/20.0);
    out_gn_l <= 10.0**((12.0-real(to_integer(unsigned(out_vol_l)))*1.5)/20.0);
  end if;
end process;
output_r <= dac_out_r*out_gn_r;
output_l <= dac_out_l*out_gn_l;</pre>
pop <= REG_20(15);
stereo_mix_r <= output_r when pop = '1' else 0.0;
stereo_mix_l <= output_l when pop = '1' else 0.0;
```

Figura 10. Registro 0x18 PCM Out Volume

Por último tan solo queda multiplicar las ganancias / atenuaciones de los registros 0x02 y 0x18 para generar las señales de salida finales en ambos canales como podemos ver en la Figura 11.

```
LINE_OUT_R <= stereo_mix_r*master_gn_r;
LINE_OUT_L <= stereo_mix_l*master_gn_l;</pre>
```

Figura 11. Salida según canal

La señal **CHANNEL** se configura cuando se pasan datos y cuando no desde el codec\_controller (recordemos la Tabla 1), dichos valores se pasan a la trama para que se pueda generar correctamente las señales de salida de ambos canales, dependiendo de los valores de la trama se pueden mutear ambos canales o pasar sólo datos por uno de ellos como podemos ver en las siguientes muestras de código de la Flgura 12:

```
master_vol_r <= REG_02(4 downto 0);
master_vol_l <= REG_02(12 downto 8);
mute_master <= REG_02(15);

process (master_vol_r, master_vol_l, mute_master) is
begin
   if mute_master = '1' then
       master_gn_r <= 0.0;
       master_gn_l <= 0.0;
   else
       master_gn_r <= 10.0**((0.0-real(to_integer(unsigned(master_vol_r)))*1.5)/20.0);
       master_gn_l <= 10.0**((0.0-real(to_integer(unsigned(master_vol_l)))*1.5)/20.0);
   end if;
end process;</pre>
```

Figura 12. Se mutean ambos o se toma valor en cada canal

Por último comentar que el siguiente pedazo de código se utiliza para saltarse 3D Sound de National tal y como se especifica en el Anexo 1 que se proporciona para realizar esta práctica, Figura 13:

```
pop <= REG_20(15);
stereo_mix_r <= output_r when pop = '1' else 0.0;
stereo_mix_l <= output_l when pop = '1' else 0.0;</pre>
```

Figura 13. Deshabilitar 3D Sound de National

#### Simulación funcional de la entidad top system

Para las simulaciones funcional y temporal cogemos el fichero *top\_system\_epp\_tb.vhd* explicado en el apartado anteriormente.

Comprobamos que nuestra entidad funciona tal como queremos, en la figura 14 estamos metiendo los valores 11x tanto en dirección como en dato, por lo que según la Tabla 1 estaríamos inicializando el códec LM4550 (**RESTART**). Después de que los datos se hayan validado con la señal **datos\_vid** vemos que se activa la señal **restart** poniendo a 0 la salida de los 2 canales.



Figura 14. Captura de simulación funcional (dirección y dato a 11x)

Comprobamos otro ejemplo mirando las figuras 15 y 16. Al igual que antes, comprobando la Tabla 1, vemos que si tenemos el valor CAx en dirección y 11x en dato vamos a transmitir audio únicamente por el canal izquierdo. En la Figura 16 vemos como al inicio de la trama, se deja de transmitir información por el canal derecho poniéndolo a 0.



Figura 15. Valores CAx en dirección y 11x en dato.



Figura 16. El canal derecho deja de transmitir información poniéndose a valor 0.

En la Figura 17 vemos un captura global de toda la simulación funcional. Primero transmitimos valores por el canal derecho, luego por los 2 canales y por último sólo transmitimos audio por el canal izquierdo.



Figura 17. Captura de la simulación funcional

#### Simulación temporal de la entidad top\_system

Con la simulación temporal comprobamos que la temporización del diseño cumple con las restricciones impuestas al mismo, fundamentalmente las relacionadas con la señal del reloj.

En la Figura 18 comprobamos que las señales resaltadas (**pwait**, **reset**, **sync** y **sdata\_out**) salen en color naranja ya que no está inicializado su valor.



Figura 18. Captura al inicio de la simulación temporal

El resto de la simulación temporal (figura 19) cumple con lo que buscábamos en el diseño de la entidad *top\_system*.



Figura 19. Captura de la simulación temporal

#### Salida MatLab

Como ya sucedía en el Apartado 2, el banco de pruebas además de la salida por pantalla de nuestra simulación nos generaba un fichero con las señales de ambos canales de audio que podíamos usar con el software matemático MatLab para verificar la salida por ambos canales de dicha simulación, en este caso nos genera la siguiente salida que vemos en la Figura 20:



Figura 20. Captura del programa Matlab

#### Informe sobre los recursos utilizados

Es condición imprescindible generar un informe (Tabla 2) sobre los recursos utilizados cuando se haga uso de la simulación temporal y posterior descarga en placa ya que de esta forma sabremos la cantidad de hardware en uso.

| Number of Slices Registers | 139 out of 54,576   | 1% |
|----------------------------|---------------------|----|
| Number of Slice LUTs       | 1,171 out of 27,288 | 4% |
| Number of occupied Slices  | 379 out of 6,822    | 5% |

Tabla 2. Recursos utilizados